查看原文
其他

千万不要这样使用Arrays.asList !

The following article is from 码上实战 Author flyhero

点击上方Java后端技术”,选择“置顶或者星标”

你关注的就是我关心的!

作者:flyhero

微信公众号:码上实战(ID:Push-Code)

使用Arrays.asList()的原因无非是想将数组或一些元素转为集合,而你得到的集合并不一定是你想要的那个集合。

而一开始asList的设计时用于打印数组而设计的,但jdk1.5开始,有了另一个比较更方便的打印函数Arrays.toString(),于是打印不再使用asList(),而asList()恰巧可用于将数组转为集合。

一、 错误用法

如果你这样使用过,那你可要注意了。

1、错误一 

将基本类型数组作为asList的参数

猜一下输出结果?

2、错误二 

将数组作为asList参数后,修改数组或List

猜一下输出结果?

3、错误三 

数组转换为集合后,进行增删元素

猜一下输出结果?

你是不是以为上面 👆那个 list 是 java.util.ArrayList ?

答案很确定:NO !

二、深入探究

我们通过asList()源码可发现其原因,但为了更直观,我们先通过IDEA debug来看看结果。

其实它返回的是  java.util.Arrays.ArrayList,这个家伙是谁呢?

请看下源码:

但它和ArrayList貌似很像唉!有什么不同吗?

三、不同之处

Arrays.ArrayList 是工具类 Arrays 的一个内部静态类,它没有完全实现List的方法,而 ArrayList直接实现了List 接口,实现了List所有方法。

  • 长度不同 和 实现的方法不同

    Arrays.ArrayList是一个定长集合,因为它没有重写add,remove方法,所以一旦初始化元素后,集合的size就是不可变的。

  • 参数赋值方式不同

Arrays.ArrayList将外部数组的引用直接通过“=”赋予内部的泛型数组,所以本质指向同一个数组。

ArrayList是将其他集合转为数组后copy到自己内部的数组的。

四、揭晓答案

1、错误一

由于Arrays.ArrayList参数为可变长泛型,而基本类型是无法泛型化的,所以它把int[] arr数组当成了一个泛型对象,所以集合中最终只有一个元素arr。

2、错误二

由于asList产生的集合元素是直接引用作为参数的数组,所以当外部数组或集合改变时,数组和集合会同步变化,这在平时我们编码时可能产生莫名的问题。

3、错误三

由于asList产生的集合并没有重写add,remove等方法,所以它会调用父类AbstractList的方法,而父类的方法中抛出的却是异常信息。

五、支持基础类型的方式

1、如果使用Spring

2、如果使用Java 8

六、数组转ArrayList

1、遍历转换

显然这种方式不够优雅!反正我不愿意使用。

2、使用工具类

上面方案不够优雅,那么这种相对来说优雅一些。

你以为这种还不错?
too young too simple!
addAll()方法的实现就是用的上面遍历的方式。

3、如果使用Java 8

既可以用于基本类型也可以返回想要的集合。

4、两个集合类结合

将Arrays.asList返回的集合作为ArrayList的构造参数

七、最后

勿以点小而不闻!体现程序素养或许就在这些小地方,不要给自己或别人留坑。那么这个知识点,你get到了吗?

热门内容:

1、2019年程序员高考真题新鲜出炉!看看你能得几分!

2、从开源小白到 Apache Member,我的成长之路!

3、GitHub 五万星登顶,命令行的艺术!

4、19 个强大、有趣、好玩、又装B的 Linux 命令!

5、如何正确访问Redis中的海量数据?服务才不会挂掉!

6、为什么有些大公司技术弱爆了?

7、必会的55个Java性能优化细节!一网打尽!

8、面试题:InnoDB中一棵B+树能存多少行数据?

您可能也对以下帖子感兴趣

文章有问题?点此查看未经处理的缓存